home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / MAIL.XPI / bin / chrome / messenger.jar / content / messenger / searchBar.js < prev    next >
Encoding:
JavaScript  |  2005-12-22  |  17.6 KB  |  558 lines

  1. /* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  * ***** BEGIN LICENSE BLOCK *****
  3.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  4.  *
  5.  * The contents of this file are subject to the Mozilla Public License Version
  6.  * 1.1 (the "License"); you may not use this file except in compliance with
  7.  * the License. You may obtain a copy of the License at
  8.  * http://www.mozilla.org/MPL/
  9.  *
  10.  * Software distributed under the License is distributed on an "AS IS" basis,
  11.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  12.  * for the specific language governing rights and limitations under the
  13.  * License.
  14.  *
  15.  * The Original Code is Mozilla Communicator client code, released
  16.  * March 31, 1998.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998-1999
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Seth Spitzer <sspitzer@netscape.com>
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either of the GNU General Public License Version 2 or later (the "GPL"),
  28.  * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. var gSearchSession = null;
  41. var gPreQuickSearchView = null;
  42. var gSearchTimer = null;
  43. var gViewSearchListener;
  44. var gSearchBundle;
  45. var gStatusBar = null;
  46. var gSearchInProgress = false;
  47. var gSearchInput = null;
  48. var gClearButton = null;
  49. var gDefaultSearchViewTerms = null;
  50. var gQSViewIsDirty = false;
  51. var gNumTotalMessages;
  52. var gNumUnreadMessages;
  53.  
  54. function SetQSStatusText(aNumHits)
  55. {
  56.   var statusMsg;
  57.   // if there are no hits, it means no matches were found in the search.
  58.   if (aNumHits == 0)
  59.     statusMsg = gSearchBundle.getString("searchFailureMessage");
  60.   else 
  61.   {
  62.     if (aNumHits == 1) 
  63.       statusMsg = gSearchBundle.getString("searchSuccessMessage");
  64.     else
  65.       statusMsg = gSearchBundle.getFormattedString("searchSuccessMessages", [aNumHits]);
  66.   }
  67.  
  68.   statusFeedback.showStatusString(statusMsg);
  69. }
  70.  
  71. // nsIMsgSearchNotify object
  72. var gSearchNotificationListener =
  73. {
  74.     onSearchHit: function(header, folder)
  75.     {
  76.       gNumTotalMessages++;
  77.       if (!header.isRead)
  78.         gNumUnreadMessages++;
  79.         // XXX todo
  80.         // update status text?
  81.     },
  82.  
  83.     onSearchDone: function(status)
  84.     {
  85.         SetQSStatusText(gDBView.QueryInterface(Components.interfaces.nsITreeView).rowCount)
  86.         statusFeedback.showProgress(0);
  87.         gStatusBar.setAttribute("mode","normal");
  88.         gSearchInProgress = false;
  89.         // ### TODO need to find out if there's quick search within a virtual folder.
  90.         if (gCurrentVirtualFolderUri &&
  91.          (gSearchInput.value == "" || gSearchInput.showingSearchCriteria))
  92.         {
  93.           var vFolder = GetMsgFolderFromUri(gCurrentVirtualFolderUri, false);
  94.           var dbFolderInfo = vFolder.getMsgDatabase(msgWindow).dBFolderInfo;
  95.           dbFolderInfo.NumUnreadMessages = gNumUnreadMessages;
  96.           dbFolderInfo.NumMessages = gNumTotalMessages;
  97.           vFolder.updateSummaryTotals(true); // force update from db.
  98.           var msgdb = vFolder.getMsgDatabase(msgWindow);
  99.           msgdb.Commit(MSG_DB_LARGE_COMMIT);
  100.         }
  101.     },
  102.  
  103.     onNewSearch: function()
  104.     {
  105.       statusFeedback.showProgress(0);
  106.       statusFeedback.showStatusString(gSearchBundle.getString("searchingMessage"));
  107.       gStatusBar.setAttribute("mode","undetermined");
  108.       gSearchInProgress = true;
  109.       gNumTotalMessages = 0; 
  110.       gNumUnreadMessages = 0;
  111.     }
  112. }
  113.  
  114. function getDocumentElements()
  115. {
  116.   gSearchBundle = document.getElementById("bundle_search");  
  117.   gStatusBar = document.getElementById('statusbar-icon');
  118.   gClearButton = document.getElementById('clearButton');
  119.   GetSearchInput();
  120. }
  121.  
  122. function addListeners()
  123. {
  124.   gViewSearchListener = gDBView.QueryInterface(Components.interfaces.nsIMsgSearchNotify);
  125.   gSearchSession.registerListener(gViewSearchListener);
  126. }
  127.  
  128. function removeListeners()
  129. {
  130.   gSearchSession.unregisterListener(gViewSearchListener);
  131. }
  132.  
  133. function removeGlobalListeners()
  134. {
  135.   removeListeners();
  136.   gSearchSession.unregisterListener(gSearchNotificationListener); 
  137. }
  138.  
  139. function initializeGlobalListeners()
  140. {
  141.   // Setup the javascript object as a listener on the search results
  142.   gSearchSession.registerListener(gSearchNotificationListener);
  143. }
  144.  
  145. function createQuickSearchView()
  146. {
  147.   //if not already in quick search view 
  148.   if (gDBView.viewType != nsMsgViewType.eShowQuickSearchResults)  
  149.   {
  150.     var treeView = gDBView.QueryInterface(Components.interfaces.nsITreeView);  //clear selection
  151.     if (treeView && treeView.selection)
  152.       treeView.selection.clearSelection();
  153.     gPreQuickSearchView = gDBView;
  154.     if (gDBView.viewType == nsMsgViewType.eShowVirtualFolderResults)
  155.     {
  156.       // remove the view as a listener on the search results
  157.       var saveViewSearchListener = gDBView.QueryInterface(Components.interfaces.nsIMsgSearchNotify);
  158.       gSearchSession.unregisterListener(saveViewSearchListener);
  159.     }
  160.     CreateDBView(gDBView.msgFolder, (gXFVirtualFolderTerms) ? nsMsgViewType.eShowVirtualFolderResults : nsMsgViewType.eShowQuickSearchResults, gDBView.viewFlags, gDBView.sortType, gDBView.sortOrder);
  161.   }
  162. }
  163.  
  164. function initializeSearchBar()
  165. {
  166.    createQuickSearchView();
  167.    if (!gSearchSession)
  168.    {
  169.      var searchSessionContractID = "@mozilla.org/messenger/searchSession;1";
  170.      gSearchSession = Components.classes[searchSessionContractID].createInstance(Components.interfaces.nsIMsgSearchSession);
  171.      initializeGlobalListeners();
  172.    }
  173.    else
  174.    {
  175.      if (gSearchInProgress)
  176.      {
  177.        onSearchStop();
  178.        gSearchInProgress = false;
  179.      }
  180.      removeListeners();
  181.    }
  182.    addListeners();
  183. }
  184.  
  185. function onEnterInSearchBar()
  186. {
  187.    if (!gSearchInput)
  188.      getDocumentElements();
  189.    viewDebug ("onEnterInSearchBar gSearchInput.value = " + gSearchInput.value + " showing criteria = " + gSearchInput.showingSearchCriteria +"\n");
  190.    if (gSearchInput.value == ""  /* || gSearchInput.showingSearchCriteria */) 
  191.    {
  192.      
  193.      if (gDBView.viewType == nsMsgViewType.eShowQuickSearchResults 
  194.         || gDBView.viewType == nsMsgViewType.eShowVirtualFolderResults)
  195.      {
  196.        statusFeedback.showStatusString("");
  197.        disableQuickSearchClearButton();
  198.  
  199.        viewDebug ("onEnterInSearchBar gDefaultSearchViewTerms = " + gDefaultSearchViewTerms + "gVirtualFolderTerms = " 
  200.         + gVirtualFolderTerms + "gXFVirtualFolderTerms = " + gXFVirtualFolderTerms + "\n");
  201.        var addTerms = gDefaultSearchViewTerms || gVirtualFolderTerms || gXFVirtualFolderTerms;
  202.        if (addTerms)
  203.        {
  204.            viewDebug ("addTerms = " + addTerms + " count = " + addTerms.Count() + "\n");
  205.            initializeSearchBar();
  206.            onSearch(addTerms);
  207.        }
  208.        else
  209.         restorePreSearchView();
  210.      }
  211.      else if (gPreQuickSearchView && !gDefaultSearchViewTerms)// may be a quick search from a cross-folder virtual folder
  212.       restorePreSearchView();
  213.      
  214. //     gSearchInput.showingSearchCriteria = true;
  215.      
  216.      gQSViewIsDirty = false;
  217.      return;
  218.    }
  219.  
  220.    initializeSearchBar();
  221.  
  222.    if (gClearButton)
  223.     gClearButton.setAttribute("disabled", false); //coming into search enable clear button   
  224.  
  225.    ClearThreadPaneSelection();
  226.    ClearMessagePane();
  227.  
  228.    onSearch(null);
  229.    gQSViewIsDirty = false;
  230. }
  231.  
  232. function restorePreSearchView()
  233. {
  234.   var selectedHdr = null;
  235.   //save selection
  236.   try 
  237.   {
  238.     selectedHdr = gDBView.hdrForFirstSelectedMessage;
  239.   }
  240.   catch (ex)
  241.   {}
  242.  
  243.   //we might have to sort the view coming out of quick search
  244.   var sortType = gDBView.sortType;
  245.   var sortOrder = gDBView.sortOrder;
  246.   var viewFlags = gDBView.viewFlags;
  247.   var folder = gDBView.msgFolder;
  248.  
  249.   gDBView.close();
  250.   gDBView = null; 
  251.  
  252.   if (gPreQuickSearchView)
  253.   {
  254.     gDBView = gPreQuickSearchView;
  255.     if (gDBView.viewType == nsMsgViewType.eShowVirtualFolderResults)
  256.     {
  257.       // readd the view as a listener on the search results
  258.       var saveViewSearchListener = gDBView.QueryInterface(Components.interfaces.nsIMsgSearchNotify);
  259.       if (gSearchSession)
  260.         gSearchSession.registerListener(saveViewSearchListener);
  261.     }
  262. //    dump ("view type = " + gDBView.viewType + "\n");
  263.  
  264.     if (sortType != gDBView.sortType || sortOrder != gDBView.sortOrder)
  265.     {
  266.       gDBView.sort(sortType, sortOrder);
  267.     }
  268.     UpdateSortIndicators(sortType, sortOrder);
  269.  
  270.     gPreQuickSearchView = null;    
  271.   }
  272.   else //create default view type
  273.     CreateDBView(folder, nsMsgViewType.eShowAllThreads, viewFlags, sortType, sortOrder);
  274.  
  275.   RerootThreadPane();
  276.    
  277.   var scrolled = false;
  278.   
  279.   // now restore selection
  280.   if (selectedHdr)
  281.   {
  282.     gDBView.selectMsgByKey(selectedHdr.messageKey);
  283.     var treeView = gDBView.QueryInterface(Components.interfaces.nsITreeView);
  284.     var selectedIndex = treeView.selection.currentIndex;
  285.     if (selectedIndex >= 0) 
  286.     {
  287.       // scroll
  288.       EnsureRowInThreadTreeIsVisible(selectedIndex);
  289.       scrolled = true;
  290.     }
  291.     else
  292.       ClearMessagePane();
  293.   }
  294.  
  295.   // NOTE,
  296.   // if you change the scrolling code below,
  297.   // double check the scrolling logic in
  298.   // msgMail3PaneWindow.js, "FolderLoaded" event code
  299.   if (!scrolled)
  300.   {
  301.     // if we didn't just scroll, 
  302.     // scroll to the first new message
  303.     // but don't select it
  304.     if (pref.getBoolPref("mailnews.scroll_to_new_message"))
  305.       scrolled = ScrollToMessage(nsMsgNavigationType.firstNew, true, false /* selectMessage */);
  306.     if (!scrolled) 
  307.     {
  308.       // if we still haven't scrolled,
  309.       // scroll to the newest, which might be the top or the bottom
  310.       // depending on our sort order and sort type
  311.       if (sortOrder == nsMsgViewSortOrder.ascending) 
  312.       {
  313.         switch (sortType) 
  314.         {
  315.           case nsMsgViewSortType.byDate: 
  316.           case nsMsgViewSortType.byId: 
  317.           case nsMsgViewSortType.byThread: 
  318.             scrolled = ScrollToMessage(nsMsgNavigationType.lastMessage, true, false /* selectMessage */);
  319.             break;
  320.         }
  321.       }
  322.       // if still we haven't scrolled,
  323.       // scroll to the top.
  324.       if (!scrolled)
  325.         EnsureRowInThreadTreeIsVisible(0);
  326.     }
  327.   }
  328.   // NOTE,
  329.   // if you change the scrolling code above,
  330.   // double check the scrolling logic in
  331.   // msgMail3PaneWindow.js, "FolderLoaded" event code
  332. }
  333.  
  334. function onSearch(aSearchTerms)
  335. {
  336.     viewDebug("in OnSearch, searchTerms = " + aSearchTerms + "\n");
  337.     RerootThreadPane();
  338.  
  339.     if (aSearchTerms)
  340.       createSearchTermsWithList(aSearchTerms);
  341.     else
  342.       createSearchTerms();
  343.  
  344.     gDBView.searchSession = gSearchSession;
  345.     try
  346.     {
  347.       gSearchSession.search(msgWindow);
  348.     }
  349.     catch(ex)
  350.     {
  351.       dump("Search Exception\n");
  352.     }
  353. }
  354.  
  355. function createSearchTermsWithList(aTermsArray)
  356. {
  357.   var nsMsgSearchScope = Components.interfaces.nsMsgSearchScope;
  358.   var nsMsgSearchAttrib = Components.interfaces.nsMsgSearchAttrib;
  359.   var nsMsgSearchOp = Components.interfaces.nsMsgSearchOp;
  360.  
  361.   gSearchSession.clearScopes();
  362.   var searchTerms = gSearchSession.searchTerms;
  363.   var searchTermsArray = searchTerms.QueryInterface(Components.interfaces.nsISupportsArray);
  364.   searchTermsArray.Clear();
  365.  
  366.   var i;
  367.   var selectedFolder = GetThreadPaneFolder();
  368.   if (gXFVirtualFolderTerms)
  369.   {
  370.     var msgDatabase = selectedFolder.getMsgDatabase(msgWindow);
  371.     if (msgDatabase)
  372.     {
  373.       var dbFolderInfo = msgDatabase.dBFolderInfo;
  374.       var srchFolderUri = dbFolderInfo.getCharPtrProperty("searchFolderUri");
  375.       viewDebug("createSearchTermsWithList xf vf scope = " + srchFolderUri + "\n");
  376.       var srchFolderUriArray = srchFolderUri.split('|');
  377.       for (i in srchFolderUriArray) 
  378.       {
  379.         var realFolderRes = GetResourceFromUri(srchFolderUriArray[i]);
  380.         var realFolder = realFolderRes.QueryInterface(Components.interfaces.nsIMsgFolder);
  381.         if (!realFolder.isServer)
  382.           gSearchSession.addScopeTerm(nsMsgSearchScope.offlineMail, realFolder);
  383.       }
  384.     }
  385.   }
  386.   else
  387.   {
  388.     viewDebug ("in createSearchTermsWithList, adding scope term for selected folder\n");
  389.     gSearchSession.addScopeTerm(nsMsgSearchScope.offlineMail, selectedFolder);
  390.   }
  391.   // add each item in termsArray to the search session
  392.  
  393.   var termsArray = aTermsArray.QueryInterface(Components.interfaces.nsISupportsArray);
  394.   for (i = 0; i < termsArray.Count(); ++i)
  395.     gSearchSession.appendTerm(termsArray.GetElementAt(i).QueryInterface(Components.interfaces.nsIMsgSearchTerm));
  396. }
  397.  
  398. function createSearchTerms()
  399. {
  400.   var nsMsgSearchScope = Components.interfaces.nsMsgSearchScope;
  401.   var nsMsgSearchAttrib = Components.interfaces.nsMsgSearchAttrib;
  402.   var nsMsgSearchOp = Components.interfaces.nsMsgSearchOp;
  403.  
  404.   // create an i supports array to store our search terms 
  405.   var searchTermsArray = Components.classes["@mozilla.org/supports-array;1"].createInstance(Components.interfaces.nsISupportsArray);
  406.   var selectedFolder = GetThreadPaneFolder();
  407.  
  408.   var searchAttrib = (IsSpecialFolder(selectedFolder, MSG_FOLDER_FLAG_SENTMAIL | MSG_FOLDER_FLAG_DRAFTS | MSG_FOLDER_FLAG_QUEUE, true)) ? nsMsgSearchAttrib.ToOrCC : nsMsgSearchAttrib.Sender;
  409.   // implement | for QS
  410.   // does this break if the user types "foo|bar" expecting to see subjects with that string?
  411.   // I claim no, since "foo|bar" will be a hit for "foo" || "bar"
  412.   // they just might get more false positives
  413.   var termList = gSearchInput.value.split("|");
  414.   for (var i = 0; i < termList.length; i ++)
  415.   {
  416.     // if the term is empty, skip it
  417.     if (termList[i] == "")
  418.       continue;
  419.  
  420.     // create, fill, and append the subject term
  421.     var term = gSearchSession.createTerm();
  422.     var value = term.value;
  423.     value.str = termList[i];
  424.     term.value = value;
  425.     term.attrib = nsMsgSearchAttrib.Subject;
  426.     term.op = nsMsgSearchOp.Contains;
  427.     term.booleanAnd = false;
  428.     searchTermsArray.AppendElement(term);
  429.  
  430.     // create, fill, and append the sender (or recipient) term
  431.     term = gSearchSession.createTerm();
  432.     value = term.value;
  433.     value.str = termList[i];
  434.     term.value = value;
  435.     term.attrib = searchAttrib;
  436.     term.op = nsMsgSearchOp.Contains; 
  437.     term.booleanAnd = false;
  438.     searchTermsArray.AppendElement(term);
  439.   }
  440.  
  441.   // now append the default view or virtual folder criteria to the quick search   
  442.   // so we don't lose any default view information
  443.   viewDebug("gDefaultSearchViewTerms = " + gDefaultSearchViewTerms + "gVirtualFolderTerms = " + gVirtualFolderTerms + 
  444.     "gXFVirtualFolderTerms = " + gXFVirtualFolderTerms + "\n");
  445.   var defaultSearchTerms = (gDefaultSearchViewTerms || gVirtualFolderTerms || gXFVirtualFolderTerms);
  446.   if (defaultSearchTerms)
  447.   {
  448.     var isupports = null;
  449.     var searchTerm; 
  450.     var termsArray = defaultSearchTerms.QueryInterface(Components.interfaces.nsISupportsArray);
  451.     for (i = 0; i < termsArray.Count(); i++)
  452.     {
  453.       isupports = termsArray.GetElementAt(i);
  454.       searchTerm = isupports.QueryInterface(Components.interfaces.nsIMsgSearchTerm);
  455.       searchTermsArray.AppendElement(searchTerm);
  456.     }
  457.   }
  458.   
  459.   createSearchTermsWithList(searchTermsArray);
  460.   
  461.   // now that we've added the terms, clear out our input array
  462.   searchTermsArray.Clear();
  463. }
  464.  
  465. function onAdvancedSearch()
  466. {
  467.   MsgSearchMessages();
  468. }
  469.  
  470. function onSearchStop() 
  471. {
  472.   gSearchSession.interruptSearch();
  473. }
  474.  
  475. function onSearchKeyPress(event)
  476. {
  477.   // 13 == return
  478.   if (event && event.keyCode == 13)
  479.     onSearchInput(true);
  480. }
  481.  
  482. function onSearchInput(returnKeyHit)
  483. {
  484.   if (gSearchTimer) {
  485.     clearTimeout(gSearchTimer); 
  486.     gSearchTimer = null;
  487.   }
  488.  
  489.   // only select the text when the return key was hit
  490.   if (returnKeyHit) {
  491.     GetSearchInput();
  492.     gSearchInput.select();
  493.     onEnterInSearchBar();
  494.   }
  495.   else {
  496.     gSearchTimer = setTimeout("onEnterInSearchBar();", 800);
  497.   }
  498. }
  499.  
  500. function onClearSearch()
  501. {
  502.   // Use the last focused element so that focus can be restored
  503.   // if it does not exist, try and get the thread tree instead
  504.   var focusedElement = gLastFocusedElement || GetThreadTree();
  505.   Search("");
  506.   focusedElement.focus();
  507. }
  508.  
  509. function disableQuickSearchClearButton()
  510. {
  511.  if (gClearButton)
  512.    gClearButton.setAttribute("disabled", true); //going out of search disable clear button
  513. }
  514.  
  515. function ClearQSIfNecessary()
  516. {
  517.   GetSearchInput();
  518.  
  519.   if (gSearchInput.value == "")
  520.     return;
  521.  
  522.   Search("");
  523. }
  524.  
  525. function Search(str)
  526. {
  527.   GetSearchInput();
  528.  
  529.   viewDebug("in Search str = " + str + "gSearchInput.showingSearchCriteria = " + gSearchInput.showingSearchCriteria + "\n");
  530.  
  531.   if (str != gSearchInput.value)
  532.   {
  533.     gQSViewIsDirty = true; 
  534.     viewDebug("in Search(), setting gQSViewIsDirty true\n");
  535.   }
  536.  
  537.   gSearchInput.value = str;  //on input does not get fired for some reason
  538.   onSearchInput(true);
  539. }
  540.  
  541. function saveViewAsVirtualFolder()
  542. {
  543.   openNewVirtualFolderDialogWithArgs(gSearchInput.value, gSearchSession.searchTerms);
  544. }
  545.  
  546. // When the front end has finished loading a virtual folder, it calls openVirtualFolder
  547. // to actually perform the folder search. We use this method instead of calling Search("") directly
  548. // from FolderLoaded in order to avoid moving focus into the search bar.
  549. function loadVirtualFolder()
  550. {
  551.   // bit of a hack...if the underlying real folder is loaded with the same view value
  552.   // as the value for the virtual folder being searched, then ViewChangeByValue
  553.   // fails to change the view because it thinks the view is already correctly loaded.
  554.   // so set gCurrentViewValue back to All. 
  555.   gCurrentViewValue = 0; 
  556.   onEnterInSearchBar();
  557. }
  558.